From: Wei Huang Date: Mon, 9 May 2011 10:39:23 +0000 (+0100) Subject: x86/fpu: create lazy and non-lazy FPU restore functions X-Git-Url: https://dgit.raspbian.org/%22http:/www.example.com/cgi/%22https:/%22bookmarks://%22Dat/%22http:/www.example.com/cgi/%22https:/%22bookmarks:/%22Dat?a=commitdiff_plain;h=5968755d6dab2b3e193bdaf1d23658d3f1328a25;p=xen.git x86/fpu: create lazy and non-lazy FPU restore functions Currently Xen relies on #NM (via CR0.TS) to trigger FPU context restore. But not all FPU state is tracked by TS bit. This function creates two FPU restore functions: vcpu_restore_fpu_lazy() and vcpu_restore_fpu_eager(). vcpu_restore_fpu_lazy() is still used when #NM is triggered. vcpu_restore_fpu_eager(), as a comparision, is called for vcpu which is being scheduled in on every context switch. To minimize restore overhead, it creates a flag, nonlazy_xstate_used, to control non-lazy restore. Signed-off-by: Wei Huang --- diff --git a/xen/arch/x86/domain.c b/xen/arch/x86/domain.c index 8c19b9e7d3..8bb0f5dd81 100644 --- a/xen/arch/x86/domain.c +++ b/xen/arch/x86/domain.c @@ -1566,6 +1566,7 @@ static void __context_switch(void) memcpy(stack_regs, &n->arch.user_regs, CTXT_SWITCH_STACK_BYTES); if ( xsave_enabled(n) && n->arch.xcr0 != get_xcr0() ) set_xcr0(n->arch.xcr0); + vcpu_restore_fpu_eager(n); n->arch.ctxt_switch_to(n); } diff --git a/xen/arch/x86/hvm/svm/svm.c b/xen/arch/x86/hvm/svm/svm.c index 0ed05cbcc1..197fa2c3a8 100644 --- a/xen/arch/x86/hvm/svm/svm.c +++ b/xen/arch/x86/hvm/svm/svm.c @@ -348,7 +348,7 @@ static void svm_fpu_enter(struct vcpu *v) { struct vmcb_struct *vmcb = v->arch.hvm_svm.vmcb; - vcpu_restore_fpu(v); + vcpu_restore_fpu_lazy(v); vmcb_set_exception_intercepts( vmcb, vmcb_get_exception_intercepts(vmcb) & ~(1U << TRAP_no_device)); } diff --git a/xen/arch/x86/hvm/vmx/vmx.c b/xen/arch/x86/hvm/vmx/vmx.c index b9aa49b599..996bbfe9a8 100644 --- a/xen/arch/x86/hvm/vmx/vmx.c +++ b/xen/arch/x86/hvm/vmx/vmx.c @@ -612,7 +612,7 @@ static int vmx_load_vmcs_ctxt(struct vcpu *v, struct hvm_hw_cpu *ctxt) static void vmx_fpu_enter(struct vcpu *v) { - vcpu_restore_fpu(v); + vcpu_restore_fpu_lazy(v); v->arch.hvm_vmx.exception_bitmap &= ~(1u << TRAP_no_device); vmx_update_exception_bitmap(v); v->arch.hvm_vmx.host_cr0 &= ~X86_CR0_TS; diff --git a/xen/arch/x86/i387.c b/xen/arch/x86/i387.c index 9350f734d5..e5780ef81b 100644 --- a/xen/arch/x86/i387.c +++ b/xen/arch/x86/i387.c @@ -98,13 +98,13 @@ static inline void fpu_frstor(struct vcpu *v) /* FPU Save Functions */ /*******************************/ /* Save x87 extended state */ -static inline void fpu_xsave(struct vcpu *v, uint64_t mask) +static inline void fpu_xsave(struct vcpu *v) { /* XCR0 normally represents what guest OS set. In case of Xen itself, * we set all accumulated feature mask before doing save/restore. */ set_xcr0(v->arch.xcr0_accum); - xsave(v, mask); + xsave(v, v->arch.nonlazy_xstate_used ? XSTATE_ALL : XSTATE_LAZY); set_xcr0(v->arch.xcr0); } @@ -160,10 +160,25 @@ static inline void fpu_fsave(struct vcpu *v) /*******************************/ /* VCPU FPU Functions */ /*******************************/ +/* Restore FPU state whenever VCPU is schduled in. */ +void vcpu_restore_fpu_eager(struct vcpu *v) +{ + ASSERT(!is_idle_vcpu(v)); + + /* save the nonlazy extended state which is not tracked by CR0.TS bit */ + if ( v->arch.nonlazy_xstate_used ) + { + /* Avoid recursion */ + clts(); + fpu_xrstor(v, XSTATE_NONLAZY); + stts(); + } +} + /* * Restore FPU state when #NM is triggered. */ -void vcpu_restore_fpu(struct vcpu *v) +void vcpu_restore_fpu_lazy(struct vcpu *v) { ASSERT(!is_idle_vcpu(v)); @@ -174,7 +189,7 @@ void vcpu_restore_fpu(struct vcpu *v) return; if ( xsave_enabled(v) ) - fpu_xrstor(v, XSTATE_ALL); + fpu_xrstor(v, XSTATE_LAZY); else if ( v->fpu_initialised ) { if ( cpu_has_fxsr ) @@ -204,7 +219,7 @@ void vcpu_save_fpu(struct vcpu *v) clts(); if ( xsave_enabled(v) ) - fpu_xsave(v, XSTATE_ALL); + fpu_xsave(v); else if ( cpu_has_fxsr ) fpu_fxsave(v); else diff --git a/xen/arch/x86/traps.c b/xen/arch/x86/traps.c index b57bbf56ca..fc4ad84202 100644 --- a/xen/arch/x86/traps.c +++ b/xen/arch/x86/traps.c @@ -3198,7 +3198,7 @@ asmlinkage void do_device_not_available(struct cpu_user_regs *regs) BUG_ON(!guest_mode(regs)); - vcpu_restore_fpu(curr); + vcpu_restore_fpu_lazy(curr); if ( curr->arch.pv_vcpu.ctrlreg[0] & X86_CR0_TS ) { diff --git a/xen/include/asm-x86/domain.h b/xen/include/asm-x86/domain.h index 095735ed85..0488e655bb 100644 --- a/xen/include/asm-x86/domain.h +++ b/xen/include/asm-x86/domain.h @@ -492,7 +492,10 @@ struct arch_vcpu * it explicitly enables it via xcr0. */ uint64_t xcr0_accum; - + /* This variable determines whether nonlazy extended state has been used, + * and thus should be saved/restored. */ + bool_t nonlazy_xstate_used; + struct paging_vcpu paging; #ifdef CONFIG_X86_32 diff --git a/xen/include/asm-x86/i387.h b/xen/include/asm-x86/i387.h index f3a7df0229..27399c0bbf 100644 --- a/xen/include/asm-x86/i387.h +++ b/xen/include/asm-x86/i387.h @@ -14,7 +14,8 @@ #include #include -void vcpu_restore_fpu(struct vcpu *v); +void vcpu_restore_fpu_eager(struct vcpu *v); +void vcpu_restore_fpu_lazy(struct vcpu *v); void vcpu_save_fpu(struct vcpu *v); int vcpu_init_fpu(struct vcpu *v);